From 7063b199b4748a9c354ed37e64cdc84c512f2c0c Mon Sep 17 00:00:00 2001 From: Armand Philippot Date: Thu, 14 Dec 2023 15:30:34 +0100 Subject: refactor(pages): rewrite helpers to output schema in json-ld format * make sure url are absolutes * nest breadcrumb schema in webpage schema * trim HTML tags from content/description * use a regular script instead of next/script (with the latter the schema is not updated on route change) * place the script in document head * add keywords, wordCount and readingTime keys in BlogPosting schema * fix breadcrumbs in search page (without query) * add tests (a `MatchInlineSnapshot` will be better but Prettier 3 is not supported yet) --- src/pages/article/[slug].tsx | 100 ++++++++++++++++++++++++------------------- 1 file changed, 56 insertions(+), 44 deletions(-) (limited to 'src/pages/article/[slug].tsx') diff --git a/src/pages/article/[slug].tsx b/src/pages/article/[slug].tsx index 6333056..e18de75 100644 --- a/src/pages/article/[slug].tsx +++ b/src/pages/article/[slug].tsx @@ -3,7 +3,6 @@ import type { ParsedUrlQuery } from 'querystring'; import type { GetStaticPaths, GetStaticProps } from 'next'; import Head from 'next/head'; import { useRouter } from 'next/router'; -import Script from 'next/script'; import { useCallback } from 'react'; import { useIntl } from 'react-intl'; import { @@ -36,11 +35,12 @@ import type { } from '../../types'; import { CONFIG } from '../../utils/config'; import { - getBlogSchema, - getCommentsSchema, - getSchemaJson, - getSinglePageSchema, - getWebPageSchema, + getBlogPostingGraph, + getCommentGraph, + getReadingTimeFrom, + getSchemaFrom, + getWebPageGraph, + trimHTMLTags, updateWordPressCodeBlocks, } from '../../utils/helpers'; import { loadTranslation, type Messages } from '../../utils/helpers/server'; @@ -129,9 +129,17 @@ const ArticlePage: NextPageWithLayout = ({ data }) => { [intl] ); + const flattenComments = useCallback( + (allComments: SingleComment[]): SingleComment[] => [ + ...allComments, + ...allComments.flatMap((comment) => flattenComments(comment.replies)), + ], + [] + ); + if (isFallback || isLoading) return ; - const { content, id, intro, meta, title } = article; + const { content, id, intro, meta, slug, title } = article; const { author, commentsCount, @@ -143,36 +151,42 @@ const ArticlePage: NextPageWithLayout = ({ data }) => { wordsCount, } = meta; - const webpageSchema = getWebPageSchema({ - description: intro, - locale: CONFIG.locales.defaultLocale, - slug: article.slug, - title, - updateDate: dates.update, - }); - const blogSchema = getBlogSchema({ - isSinglePage: true, - locale: CONFIG.locales.defaultLocale, - slug: article.slug, - }); - const blogPostSchema = getSinglePageSchema({ - commentsCount, - content, - cover: cover?.src, - dates, - description: intro, - id: 'article', - kind: 'post', - locale: CONFIG.locales.defaultLocale, - slug: article.slug, - title, - }); - const schemaJsonLd = getSchemaJson([ - webpageSchema, - blogSchema, - blogPostSchema, - breadcrumbSchema, - ...getCommentsSchema(comments), + const jsonLd = getSchemaFrom([ + getWebPageGraph({ + breadcrumb: breadcrumbSchema, + copyrightYear: new Date(dates.publication).getFullYear(), + dates, + description: trimHTMLTags(intro), + slug, + title, + }), + getBlogPostingGraph({ + body: trimHTMLTags(content), + comment: flattenComments(comments).map((comment) => + getCommentGraph({ + articleSlug: slug, + author: { + '@type': 'Person', + name: comment.meta.author.name, + url: comment.meta.author.website, + }, + body: trimHTMLTags(comment.content), + id: `${comment.id}`, + parentId: comment.parentId ? `${comment.parentId}` : undefined, + publishedAt: comment.meta.date, + }) + ), + commentCount: commentsCount, + copyrightYear: new Date(dates.publication).getFullYear(), + cover: cover?.src, + dates, + description: trimHTMLTags(intro), + keywords: topics?.map((topic) => topic.name).join(', '), + readingTime: `PT${getReadingTimeFrom(wordsCount).inMinutes()}M`, + slug, + title, + wordCount: meta.wordsCount, + }), ]); const pageUrl = `${CONFIG.url}${article.slug}`; @@ -200,14 +214,12 @@ const ArticlePage: NextPageWithLayout = ({ data }) => { +